home *** CD-ROM | disk | FTP | other *** search
/ Technotools / Technotools (Chestnut CD-ROM)(1993).ISO / lang_pas / io5150 / io5150.pas
Pascal/Delphi Source File  |  1987-02-04  |  10KB  |  307 lines

  1. {
  2.      Device driver for IBM 5150 PC's serial port (AUX).
  3.      By J. Eric Roskos
  4.      Public Domain, may not be sold for profit.
  5.  
  6.      This driver is initialized by calling AuxInit, and MUST be
  7.      deactivated by calling AuxOff before the program exits (or a sub-
  8.      sequent serial interrupt will cause the system to hang).
  9.      It uses serial interrupts, and thus works considerably better than
  10.      the standard AUX device driver supplied with the IBM PC.
  11.  
  12.      Once AuxInit is called, you can read and write to the predefined
  13.      Turbo Pascal file "Aux" to access the serial port.  The procedure
  14.      AuxSt returns True if a character is presently available at the
  15.      serial port, equivalent to the KeyPressed procedure for the console.
  16.      You don't have to (and shouldn't) call any of the other procedures
  17.      in this file other than AuxInit, AuxOff, and AuxSt.
  18.  
  19.      Presently, only input is done via interrupts; output is via
  20.      conventional polling, and a write to the serial port will cause
  21.      a busy-wait until any previous character is completely transmitted.
  22.      The input buffer's size is defined by the constant BUFSIZ below,
  23.      which must be a power of 2 (or the wrap-around algorithm won't work).
  24.      The serial port is presently initialized to 1200 baud; if you want
  25.      to set it via the MODE command of DOS instead, you can comment out
  26.      the statements indicated below.
  27.  
  28.      If the input buffer becomes full, a ^S is sent to the remote machine;
  29.      a ^Q is then sent when it empties sufficiently to resume.  This
  30.      is indeed used at 1200 baud if a number of escape sequences are
  31.      sent in succession.
  32.  
  33.      Only the top 24 lines of the screen are used, in order to comply with
  34.      most standard terminals.  However, for this to work you have to
  35.      compile the program with Turbo Pascal version 2.0 or later, and use
  36.      an IBM PC version of the compiler (not the generic MS-DOS version).
  37. }
  38.  
  39. (*
  40. unit IO5150;
  41. interface
  42.      procedure AuxInit;     { Initialize the unit; must be called at start }
  43.      procedure AuxOff;      { Terminates the unit; must be called at exit  }
  44.      procedure AuxSt;       { Returns True if char immed. avail from Aux   }
  45.      procedure AuxParm;     { Sets baud and parity }
  46.  
  47. implementation
  48. *)
  49.  
  50. {$R-}
  51. {$C-}
  52.  
  53. const
  54.      BUFSIZ = 1024;    { Buffer size -- must be a power of 2 }
  55.                        { buffer size raised to 1024 by JP }
  56.  
  57.      { Port addresses for COM1: you must change these to use COM2 }
  58.  
  59.      RX     = 1016;   { Receiver Buffer Register            }
  60.      TX     = 1016;   { Transmitter Buffer Register         }
  61.      IE     = 1017;   { Interrupt Enable Register           }
  62.      II     = 1018;   { Interrupt Identification Register   }
  63.      LC     = 1019;   { Line Control Register               }
  64.      MC     = 1020;   { Modem Control Register              }
  65.      LS     = 1021;   { Line Status Register                }
  66.      MS     = 1022;   { Modem Status Register               }
  67.      DLL    = 1016;   { Divisor Latch, Low Order Byte       }
  68.      DLH    = 1017;   { Divisor Latch, High Order Byte      }
  69.  
  70.  
  71. type
  72.      { The input buffer structure }
  73.  
  74.      buffer=record
  75.           buf: packed array[0..BUFSIZ] of char; { The character buffer       }
  76.           ip,op,cnt: integer; { Input pointer, output pointer, char count    }
  77.      end;
  78.  
  79.      { Turbo Pascal's DOS/BIOS call parameter block }
  80.  
  81.      regpack=record
  82.           ax,bx,cx,dx,bp,si,di,ds,es,flags: integer;
  83.      end;
  84.  
  85.  
  86. var
  87.      Buf:    buffer;                       { The input buffer                }
  88.      AuxOset, AuxSeg: integer;             { Saved orig. serial int vector   }
  89.      SvOset: integer absolute $0000:$0030; { The serial interrupt vector     }
  90.      SvSeg:  integer absolute $0000:$0032; {  "    "        "       "        }
  91.      SavDs:  integer absolute $0000:$002E; { Program's DS addr is saved here }
  92.      Run:    boolean;                      { True while emulator is to run   }
  93.      XedOff: boolean;                      { True if IntSer Xed off remote   }
  94.      c:      char;                         { Current input character         }
  95.  
  96.  
  97. {
  98.      Interrupt service routine.
  99.  
  100.      This routine saves all registers, and then sets up its Data Segment
  101.      from the value saved in 0:$2E by AuxInit (since, contrary to what the
  102.      Turbo manual tells you, the DS may be changed by the ROM BIOS and thus
  103.      may be incorrect on entry to the interrupt routine).  It then gets the
  104.      input character (if the interrupt was due to a received character) and
  105.      stores it in the buffer, handling XOFFs if necessary.
  106. }
  107.  
  108. procedure IntSer;
  109. var
  110.      i: integer;
  111.  
  112. begin
  113.      { Save all registers and set up our Data Segment }
  114.  
  115.      inline($50/$53/$51/$52/$57/$56/$06/$1e);
  116.      inline($06/$50/$B8/$00/$00/$8e/$C0/$26/$8E/$1E/$2E/$00/$58/$07);
  117.  
  118.      { Process the interrupt }
  119.  
  120.      case Port[II] of
  121.      0:  {   i := Port[MS]  };                     { Modem Status Intr }
  122.      1:   ;                                        { No Interrupt      }
  123.      2:   writeln('Error: THRE interrupt');        { Transmit Intr     }
  124.      4:                                            { Receive Intr      }
  125.           begin
  126.              {  writeln('int'); }
  127.                Buf.buf[Buf.ip] := chr(Port[RX]);
  128.                Buf.ip := (Buf.ip + 1) and (BUFSIZ-1);
  129.                inline($FA); { CLI }
  130.                Buf.cnt := Buf.cnt + 1;
  131.                inline($FB); { STI }
  132.                if (Buf.cnt >= BUFSIZ-25) and not Xedoff then
  133.                begin
  134.                     Xedoff := true;
  135.                     Port[TX] := ord(^S);
  136.                end;
  137.           end;
  138.      6:   { i := Port[LS] } ;                        { Line Status Intr  }
  139.      end {case};
  140.  
  141.      { Turn interrupts back on }
  142.  
  143.      inline($FB);
  144.      Port[$20] := $20;
  145.  
  146.      { Restore saved registers and do IRET }
  147.  
  148.      inline($1F/$07/$5E/$5F/$5A/$59/$5B/$58/$8B/$E5/$5D/$CF);
  149. end;
  150.  
  151.  
  152. {
  153.      AUX port status.  Returns True if there is a character available
  154.      for reading, same as KeyPressed for console.
  155. }
  156.  
  157. function AuxSt: boolean;
  158. begin
  159.      AuxSt := (Buf.cnt > 0);
  160. end;
  161.  
  162.  
  163. {
  164.      AUX input routine.  Not called by the user: called by the Turbo
  165.      runtime system when you do a read from the Aux file.
  166. }
  167.  
  168. function AuxIn: char;
  169. begin
  170.      while Buf.cnt = 0 do ;
  171.  
  172.      AuxIn := Buf.buf[Buf.op];
  173.      Buf.op := (Buf.op + 1) and (BUFSIZ-1);
  174.  
  175.      inline($fa); { CLI }
  176.      Buf.cnt := Buf.cnt - 1;
  177.      inline($fb); { STI }
  178.  
  179.      if (Buf.cnt < 25) and Xedoff then
  180.      begin
  181.           Xedoff := false;
  182.           Port[TX] := ord(^Q);
  183.      end;
  184. end;
  185.  
  186.  
  187. {
  188.      AUX port output routine.  Not called by the user: called by the
  189.      Turbo runtime system when you do a write to the Aux file.
  190. }
  191.  
  192. procedure AuxOut(c:char);
  193. begin
  194.      while (Port[LS] and $20) = 0 do ; { Busy Wait }
  195.      Port[TX] := ord(c);
  196.  
  197. end;
  198.  
  199.  
  200. {
  201.      AUX device driver initialization.  You must call this before accessing
  202.      the Aux file if you want to use these device drivers.
  203. }
  204.  
  205. procedure AuxInit;
  206. var intmask, dummy:integer;
  207. begin
  208.  
  209.      inline($fa); { CLI }
  210.  
  211.      { Initialize interrupt vector and 8259A }
  212.  
  213.      SavDs     := Dseg;
  214.      AuxOset   := SvOset;
  215.      AuxSeg    := SvSeg;
  216.      SvOset    := ofs(IntSer);
  217.      SvSeg     := Cseg;
  218.      intmask   := 16;                   { interrupt controller mask bit }
  219.      Port[$21] := ( Port[$21] and (not intmask)); { enable ints 8, 9, C, E }
  220.  
  221.      { Initialize 8250 UART }
  222.      { Comment the starred lines out if you want to use MODE to init AUX: }
  223.  
  224.      Port[MC]  := $00;  { *** } { Negate DTR and RTS             }
  225.      Port[LC]  := $80;  { *** } { Set baud rate to 1200 baud     }
  226.      Port[DLL] := $60;  { *** } { Use $80 here for 300 baud      }
  227.      Port[DLH] := $00;  { *** } { Use $01 here for 300 baud      }
  228.      Port[LC]  := $2a;  { *** } { Set stick parity 7 data 1 stop }
  229.      Port[IE]  := $01;          { Turn on received data interrupts }
  230.      Port[MC]  := $0b;          { Turn on int gate, DTR, and RTS }
  231.  
  232.  
  233.      { Initialize buffer }
  234.  
  235.      Buf.ip  := 0;
  236.      Buf.op  := 0;
  237.      Buf.cnt := 0;
  238.      Xedoff  := false;
  239.  
  240.      { Initialize I/O system }
  241.  
  242.      AuxInPtr  := ofs(AuxIn);
  243.      AuxOutPtr := ofs(AuxOut);
  244.  
  245.      dummy := port[RX];   { do this read to allow other ints through }
  246.  
  247.      inline($fb); { STI }
  248. end;
  249.  
  250.  
  251. {
  252.      Reset the AUX port to non-interrupt mode.  You MUST call this routine
  253.      before you exit.
  254. }
  255.  
  256. procedure AuxOff;
  257. begin
  258.      inline($fa); { CLI }
  259.  
  260.      SvOset := AuxOset;{ Restore serial int vector to its original value  }
  261.      SvSeg  := AuxSeg;
  262.  
  263.      Port[IE]  := $00; { Turn off UART's interrupt enables                }
  264.      Port[MC]  := $00; { Negate RTS and CTS, and turn off interrupt gate  }
  265.      Port[$21] := $BC; { Set 8259 PIC back to disabling serial interrupts }
  266.      Port[$20] := $20; { Send EOI to PIC                                  }
  267.  
  268.      inline($fb); { STI }
  269. end;
  270.  
  271.  
  272. {
  273.      Change the speed and parity of the communication line.
  274.  
  275.      baud must be 300, 1200, or 2400.
  276.      parity must be E, O, or N for none (and 8 data bits)
  277.      The parity must be in uppercase!
  278.  
  279.  
  280.      Jeff Porter
  281.      6-12-85
  282.  
  283. }
  284.  
  285. procedure AuxParm(baud:integer; parity:char);
  286. var bcod, pcod : integer;
  287.     intdat : regpack;
  288. begin
  289.  
  290.  
  291.      pcod := 0;
  292.      bcod := 0;
  293.      case baud of
  294.      300:    pcod := $40;    { 010x xxxx binary }
  295.     1200:    pcod := $80;    { 100x xxxx binary }
  296.     2400:    pcod := $a0;    { 101x xxxx binary }
  297.     end; {case}
  298.     case parity of
  299.     'E' :    bcod := $1a;    { xxx1 1010 binary }
  300.     'O' :    bcod := $0a;    { xxx0 1010 binary }
  301.     'N' :    bcod := $03;    { xxx0 0011 binary }
  302.     end; {case}
  303.     if ( (pcod>0) and (bcod>0) ) then
  304.        begin { both were defined so change baud }
  305.           intdat.ax := pcod+bcod; { put 0 in AH, setting in AL }
  306.           intdat.dx := 0;         { for com1:; must change for com2: }
  307.